Questions

# C to ARM Questions

## F18. C to ARM Points: \_\_\_/20

Convert the following snippet of C-code to ARM assembly. Assume X30 holds the memory address of the instruction you should return to after the function finishes.

**Hints:** \*\* Register assignments:

x -> X1

&(arr[x]) -> X3

arr[x].d -> X6

\*\* The size of struct bar (with padding) is 24 bytes.

\*\*XZR is a register with value 0

|  |  |  |
| --- | --- | --- |
| **C-code** |  | **ARM assembly** |
| #include <inttypes.h>  struct Bar {  char a; // 1B  int64\_t b; // 8B size int64\_t\* c; // 4B size int16\_t d; // 2B size  };  void foo() {  struct Bar arr[8];  int32\_t x = 0;  for (; x < 8; x++) {  arr[x].d = x \* 16;  if (arr[x].d == 32  || arr[x].d == 64) {  arr[x].a = 0;  }  }  return;  } | 0  1  2  3  4  5  6  7  8  9  10  11  12  13  14  15  16  17  18  19  20  21 | Foo MOVI X1, #0  MOV X3, arr  loop CMPI \_\_\_\_, #\_\_\_\_  B.GE \_\_\_\_\_  LSL X6, \_\_\_\_, #4  \_\_\_\_ \_\_\_\_, [\_\_\_\_, #20]  CMP X6, #32  B.EQ \_\_\_\_  CMP X6, #\_\_\_\_  B.NE \_\_\_\_\_\_  true \_\_\_\_\_\_ XZR, [\_\_\_\_, #\_\_\_\_] endif ADDI X3, X3, #\_\_\_\_  ADDI X1, X1, #\_\_\_\_  B \_\_\_\_\_\_  done BR X30 |

## F20: C-to-ARM Points: \_\_\_/14

Convert the following C program into ARM assembly by filling in the blanks. The function **isFound** should be ABI compliant and return true(1) or false(0) in register X0.

Assume X0 contains the starting address of addr\_book[]. The program returns 1 if an address is found in the address book (addr\_book).

|  |  |
| --- | --- |
| **C** | **ARM** |
| struct House {  int32\_t num; //4B  char\* street; //8B  };  int32\_t  **isFound**(struct House addr\_book[]) {  char\* foo\_street = 0xBEAD;  int32\_t foo\_num = 370;  for(int32\_t i = 0; i < 256; i++) {  if(addr\_book[i].num == foo\_num &&  addr\_book[i].street == foo\_street)  {  return 1;  }  }  return 0;  } | isFound:  MOVZ X1, #0xBEAD, LSL 0  MOVZ X2, #370, LSL 0  MOVZ X3, \_\_\_\_, LSL 0  loop:  CMPI X3, \_\_\_\_  B.EQ ret0  LSL X4, X3, \_\_\_\_  ADD X7, X4, \_\_\_\_  LDURSW X5, [X7, \_\_\_\_]  CMP \_\_\_\_, X2  B.NE cont  LDUR X6, [X7, \_\_\_\_]  CMP \_\_\_\_, \_\_\_\_  B.EQ \_\_\_\_  cont:  ADDI \_\_\_\_, X3, #1  B loop  ret1:  MOVZ X0, \_\_\_\_, LSL 0  B end  ret0:  MOVZ \_\_\_\_, #0, LSL 0  end: |

## W18: Get Your C LEGv8’s Points: \_\_\_/ 15

Consider the following C code for (virtual) sailing. Finish converting the body of the function to assembly using the LEGv8 subset of ARM by filling out the blanks in the code provided. Assume that the system is **little-endian**. Assume that the arguments to sail are passed in via the specified registers, and that the address of wind is preloaded into register x3. Use the register assignments specified in the comments below.

x0 = lat\_dest x1 = fuel x2 = lat\_now x3 = wind

|  |  |
| --- | --- |
| C | LEGv8 |
| #include <stdint.h>  struct {  int32\_t time;  int64\_t lat\_speed;  } wind;    sail(int64\_t lat\_dest,  int64\_t fuel) {  int lat\_now = 0;    while (lat\_now != lat\_dest) {  lat\_now += wind.lat\_speed;  if (fuel > 0) {  if (lat\_now <= lat\_dest) {  lat\_now += 1;  }  else {  lat\_now -= 1;  }  --fuel;  }  wind.time += 1;  }  } | mov x2, #0  b while\_cond  loop: ldur x4, [x3, **\_\_\_\_**]  add x2, x2, x4  cmp x1, **\_\_\_**  b.gt **\_\_\_\_\_\_\_\_**  b **\_\_\_\_\_\_\_\_**  has\_fuel: cmp x0, x2  b.lt lat\_bigger  addi x2, x2, #1  b **\_\_\_\_\_\_\_\_**  lat\_bigger: addi x2, x2, #-1  dec\_fuel: addi x1, x1, #-1  time\_inc: ldursw x4, [x3, #0]  addi x4, x4, #1  sturw x4, [x3, #0]  while\_cond: cmp x0, **\_\_\_**  b.ne loop |

## W20: C-to-ARM: I’ll Give You An ARM Points: \_\_\_/ 15

Convert the following C program into ARM assembly. Your function calcBMI should be ABI compliant and return BMIs in register X0.

Hints:

// X0 - contains starting address of arr[]

// X1 - contains starting address of BMIs[]

// X2 - holds i

UDIV Rd, Rn, Rm // Unsigned divide: Rd = Rn / Rm

What is the size of struct Person in bytes? \_\_\_\_\_\_\_\_

|  |  |
| --- | --- |
| **C** | **ARM** |
| struct Person {  char name[10]; // 10B  int32\_t age; // 4B  int64\_t weight; // 8B  int64\_t height; // 8B  };  int32\_t \*  **calcBMI**(struct Person arr[],  int32\_t BMIs[]) {  int32\_t i = 0;  for(; i < 100; i++) {  if(arr[i].age > 0) {  BMIs[i] = (int32\_t)arr[i].weight/  (arr[i].height\*arr[i].height);  }  }  return BMIs;  } | calcBMI:  MOVZ X2, #0  loop:  CMPI X2, #\_\_\_\_  \_\_\_\_ end  LSL X7, X2, #\_\_\_\_  ADD X3, X0, X7  \_\_\_\_ X4, [\_\_\_\_, #\_\_\_\_]  CMPI X4, #0  \_\_\_\_ inc  true:  LSL X7, X2, #\_\_\_\_  ADD X4, \_\_\_\_, X7  LDUR X5, [X3, #16 ]  LDUR X6, [X3, #\_\_\_\_]  MUL X6, \_\_\_\_, X6  UDIV X5, X5, \_\_\_\_  \_\_\_\_ X5, [X4, #\_\_\_\_]  inc:  ADDI X2, \_\_\_\_, #1  B loop  end:  MOV \_\_\_\_, X1 //return |

## W21: C-to-ARM and the SODA370 factory

The SODA370 factory produces soda packs with a multiple of four soda cans in each pack. However, some packing machines are starting to malfunction, making some soda packs contain a number of soda cans not divisible by four. Someone has written a quality check function below in C to check how many soda packs contain incorrect amounts of soda cans. Convert part of this function into ARM.

Assume X0 contains the starting address of data[]. The function **quality\_check** returns the amount of bad soda packs in register X2.

|  |  |
| --- | --- |
| **C** | **ARM** |
| struct soda\_pack {  char pack\_id[4]; //4B  int64\_t num\_cans; //8B    };  int64\_t  **quality\_check**(struct soda\_pack data[]) {  int32\_t bad\_packs = 0;  for(int32\_t i = 0; i < 256; i++) {  if(data[i].num\_cans % 4 != 0){  bad\_packs++;  }  }  return bad\_packs;  } | qcheck:  MOVI X2, #0x00  MOVI X3, #0x00  loop:  CMPI X3, \_\_\_  B.\_\_ \_\_  \_\_\_\_ X4, \_\_\_, \_\_\_  \_\_\_\_ X5, \_\_\_, \_\_\_  LDUR X6, [\_\_\_, \_\_\_]  \_\_\_\_ X7, \_\_\_, \_\_\_  CMPI \_\_\_, #0  B.EQ good  bad:  ADDI X2, #1  good:  ADDI X3, #1  B loop  end: |

## W22: Convert C to ARM Assembly [12 Points]

Convert this C code to ARM assembly by completing the ***twelve missing instruction parts***. Registers **X0** and **X1** hold the starting addresses of **set1** and **set2**, while registers **X2**, and **X3** hold the length of each set, respectively. The input argument of **search\_value** is mapped to **X5** andreturn value of **find\_mutual\_sum** is at **X7**. Note that both set1 and set2 include **signed values**.

|  |  |  |  |  |  |
| --- | --- | --- | --- | --- | --- |
| **C** | **ARM** | | | | |
| **int64\_t set1[] = [1, 5, -3, -8, …];**  **int32\_t set2[] = [-3, 2, -4, 5, …];**  **int64\_t len1; // length of set1**  **int64\_t len2; // length of set2**  **int64\_t search\_value**(int64\_t input);  **int64\_t find\_mutual\_sum**()  {  int64\_t sum = 0;  int idx1 = 0;  for ( ; idx1 < len1 ; idx1++) {  int64\_t input = set1[idx1];  if (**search\_value**(input) > 0)  sum += input;  }  return sum;  }  **int64\_t search\_value**(int64\_t input) {  int idx2 = len2 - 1;  for ( ; idx2 >= 0 ; idx2--) {  int64\_t search =  **(int64\_t)** set2[idx2];    if (search == input)  return 1;  }  return 0;  } | **FIND\_MUTUAL\_SUM:** | | | |  |
|  | MOVZ | X7, | #0, | LSL 0 |
|  | MOVZ | X10, | #0, | LSL 0 |
| **FIND\_LOOP:** | | | |  |
|  | CMP | X10, | X2 |  |
|  | B.GE | END\_FIND\_LOOP | | |
|  | LSL | X11, | **\_\_\_\_\_**, | **\_\_\_\_\_** |
|  | ADD | X11, | X11, | X0 |
|  | LDUR | X5, | [X11, | #0] |
|  | **\_\_\_\_\_** | SEARCH\_VALUE | | |
|  | CBZ | **\_\_\_\_\_**, | CONT\_FIND\_LOOP | |
|  | ADD | X7, | X7, | **\_\_\_\_\_**, |
| **CONT\_FIND\_LOOP:** | | |  |  |
|  | ADDI | X10, | X10, | #1 |
|  | B | FIND\_LOOP | |  |
|  | |  |  |  |
| **SEARCH\_VALUE:** | |  |  |  |
|  | MOVZ | X8, | **\_\_\_\_\_**, | LSL 0 |
|  | **\_\_\_\_\_** | X12, | X3, | #1 |
| **SEARCH\_LOOP:** | |  |  |  |
|  | CMP | X12, | XZR |  |
|  | **\_\_\_\_\_** | Return |  |  |
|  | LSL | X13, | X12, | **\_\_\_\_\_** |
|  | ADD | X13, | X13, | X1 |
|  | **\_\_\_\_\_** | X6, | [X13, | #0] |
|  | CMP | X6, | **\_\_\_\_\_** |  |
|  | B.EQ | IF\_ELSE\_SEARCH | |  |
|  | SUBI | X12, | X12, | #1 |
|  | B | SEARCH\_LOOP | |  |
| **IF\_ELSE\_SEARCH** | | |  |  |
|  | ADDI | X8, | XZR, | #1 |
| **Return:** | |  |  |  |
|  | **\_\_\_\_\_** | X30 |  |  |
| **END\_FIND\_LOOP:** | | |  |  |

# C to LC2K Questions

## F19: Can you C LC2K? [16 points]

Below is a function written in C and an LC2K assembly function which does the same thing.

int main() {

letao(5);

}

int letao(int n) {

int a;

if (n == 0) {

return 1;

}

a = letao(n-1);

a = a + a;

return a;

}

main lw 0 1 value

lw 0 4 func

lw 0 2 one

lw 0 5 neg1

jalr 4 7

halt

letao beq 1 0 base

sw 6 7 Stack

add 6 2 6

add 1 5 1

jalr 4 7

add 6 5 6

lw 6 7 Stack

add 3 3 3

jalr 7 4

base add 0 2 3

jalr 7 4

func .fill letao

one .fill 1

neg1 .fill -1

value .fill 5

Stack .fill 0

.fill 0

.fill 0

.fill 0

For each register, write the letter which best describes what that register is used for. You may use a given letter more than once. [2 points each, no partial credit

R0: \_\_\_\_\_\_\_\_\_

R1: \_\_\_\_\_\_\_\_\_

R2: \_\_\_\_\_\_\_\_\_

R3: \_\_\_\_\_\_\_\_\_

R4: \_\_\_\_\_\_\_\_\_

R5: \_\_\_\_\_\_\_\_\_

R6: \_\_\_\_\_\_\_\_\_

R7: \_\_\_\_\_\_\_\_\_

A. n

B. 0

C. a

D. Stack pointer

E. -1

F. Return address

G. 5

H. letao’s address

I. 1

J. n+2a

## W18: Identity Matrix Points: \_\_\_/ 15

Consider the MATLAB function eye(N), which generates a NxN matrix with ones along the diagonal and the rest of the matrix values as zero.

Example: eye(3) matrix printed below.

1 0 0

0 1 0

0 0 1

Implement this function in LC2K by completing the blanks below. Instead of a 2D array the matrix is represented as a 1D array called eye of size N\*N (See example in C code). The final result of eye(3) should look like the comment in the C code. Use the registers naming as shown:

R0 = 0 R1 = 1 R2 = N

R3 = i R4 = j R5 = k + starting address of EYE

|  |  |
| --- | --- |
| // const int N = 3;  // int eye[N\*N] = {1, 0, 0,  // 0, 1, 0,  // 0, 0, 1};  // eye(N) function in C code:  const int N = 3;  int eye[N\*N];  for(int k=0; k<N\*N; ++k) {  eye[k] = 0;  }  **//LC2K implementation starts here**  int k = 0;  for(int i=0; i<N; ++i) {  for(int j=0; j<N; ++j) {  if(i == j) {  eye[k] = 1;  }  ++k;  }  }  **//LC2K implementation ends here** | lw 0 5 EYE  lw 0 1 ONE  lw 0 2 N  lw 0 3 ZERO  BEG\_I beq **\_** 2 END\_I  lw 0 4 ZERO  BEG\_J beq **\_ \_** END\_J  beq 3 4 **\_\_\_\_\_**  ADD\_K add **\_ \_** 5  add 4 1 4  beq 0 0 **\_\_\_\_\_**  END\_J **\_\_\_** 3 1 3  beq 0 0 BEG\_I  SET1 **\_\_\_** 5 1 0  beq 0 0 ADD\_K  END\_I halt  EYE .fill 100  ONE .fill 1  N .fill 3  ZERO .fill 0 |

## W20: LC2K Assembly Points: \_\_\_/20

Convert the following C-code to LC2K assembly. **Assume all registers are initialized to 0.** Hint. Less than (<) operation is not supported natively in LC2K. However a < b can be done in LC2K by subtracting b from a, using the sign bit of the result and comparing the sign bit to 0.

|  |  |  |
| --- | --- | --- |
| **C-code** |  | **LC2K assembly** |
| int size = 10; //size --> R2  int arr1[5] =  {1, 3, 5, 7, 9};  int arr2[5] =  {2, 4, 6, 8, 10};  int dest[10];  int ind1 = 0, ind2 = 0;  //ind1 --> R3, ind2 --> R4  int num = 0; //num --> R5  while (num != size)  {  if (arr1[ind1]<arr2[ind2]) {  dest[num]= arr1[ind1];  ind1++;  num++;  } else {  dest[num]= arr2[ind2];  ind2++;  num++;  }  } | 0  1  2  3  4  5  6  7  8  9  10  11  12  13  14  15  16  17  18  19  20  21  22  23  24  25  26  27  28  29  30  31  32  33  34  35 | lw 0 1 one  lw 0 2 size  while beq 5 2 end  lw \_\_\_ 6 arr1  lw \_\_\_ 7 arr2  nor 0 \_\_\_ 7  add 1 7 7  add 6 \_\_\_ 6  lw 0 7 mask  nor \_\_\_ \_\_\_ 6  beq 6 0 \_\_\_\_\_\_  lw 4 6 arr2  sw \_\_\_\_ 6 dest  add 1 4 4  add 1 5 5  beq 0 0 \_\_\_\_\_\_  less lw 3 \_\_\_ arr1  sw 5 7 \_\_\_\_\_\_  add 1 \_\_\_ 3  add 1 5 5  beq 0 0 \_\_\_\_\_\_\_  end halt  one .fill 1  size .fill 10  mask .fill 2147483647 #0x7FFFFFFF arr1 .fill 1  .fill 3  .fill 5  .fill 7  .fill 9  arr2 .fill 2  .fill 4  .fill 6  .fill 8  .fill 10  dest .fill 0 |

## W21: C to LC2K

*This question was removed from the final version of the exam, but has been available in practice exams since.*

1. Convert C code to LC2K by filling the blanks.

Note this does not follow the standard LC2K ABI:

reg2 has the start address of arr.

reg4 is the input real of find\_number.

reg5 is the stack pointer.

|  |  |
| --- | --- |
| **C** | **LC2K** |
| struct number{  int real;  int imagine;  }  struct number arr[5];  extern int find\_number(struct number arr[], int idx, int real)  int main(){  find\_number(arr, 4, 3);  } | main.as  lw 0 2 \_\_\_\_\_\_  lw 0 3 size  lw 0 6 neg1  lw 0 4 real  add 3 6 3  lw 0 6 \_\_\_\_\_\_\_  jalr 6 1  halt  arr .fill Array  size .fill 5  neg1 .fill -1  real .fill 3  Array .fill 3  .fill 3  .fill 2  .fill 3  .fill 3  .fill 5  .fill 8  .fill 3  .fill 5  .fill 0 |

|  |  |
| --- | --- |
| **C** | **LC2K** |
| struct number{  int real;  int imagine;  }  int find\_number(struct number arr[], int idx, int real){  if (arr[idx].real == real){  return idx;  }  else if(idx == 0){  return -1;  }  else{  return find\_number(arr, idx-1, real)  }  } | Func add \_ \_ 7 //step1: arr[idx]  add 7 2 7 //step2: arr[idx]  lw 7 7 0 //step3: arr[idx]  beq 7 4 \_\_\_\_\_\_  beq 3 0 \_\_\_\_\_\_  lw 0 6 neg1  add 6 3 3  lw 0 6 \_\_\_\_\_\_  lw 0 7 one  sw 5 1 Stack  add 7 5 5  jalr \_ \_  lw 0 7 neg1  add 7 5 5  lw 5 1 Stack  beq 0 0 \_\_\_\_\_\_  noHit lw 0 3 neg1  ret jalr \_ \_  Faddr .fill Func  neg1 .fill -1  one .fill 1 |

1. What is register 3 used for? (select all apply)

A. return address

B. idx

C. arr

D. real

E. Stack pointer

F. return\_value

G. temporary register

1. Which register has the return address? Is this register a caller save register or a callee save register? (select all apply)
2. r1
3. r2
4. r3
5. r4
6. r5
7. r6
8. r7
9. Caller save register
10. Callee save register

4. What is the value that find\_number returns?

# C to Other Questions

## F18. New ISA Points: \_\_\_/25

In 982 years, the College of Engineering decides to release its next generation ISA, the LC3K. LC3K features the following instruction types:

|  |  |  |  |
| --- | --- | --- | --- |
| **Type** | **Instruction Name** | **Opcode** | **Function** |
| **R-type** | add | 000 | Add contents of regA with, contents of regB, store results in destReg. |
| **R-type** | nand | 001 | NAND contents of regA with contents of regB store results in destReg. This is a bitwise operation; each bit is treated independently. |
| **I-type** | bgt | 010 | If the contents of regA is greater than the contents of regB then branch to the address PC+1+offsetField, where PC is the address of this BGT instruction. |
| **I-type** | lw | 011 | Load regB from memory. Memory address is formed by adding offsetField with the contents of regA. |
| **I-type** | sw | 100 | Store regB into memory. Memory address is formed by adding offsetFIeld with the contents of regA. |
| **S-type** | shift | 101 | Shift the contents in regA to the left (positive) or right(negative) by the number of bits in the offsetField. |
| **K-type** | swap | 110 | Swap the contents of regA and regB. |
| **O-type** | halt | 111 | Increment the PC (as with all instructions), then halt the machine (let the simulator notice that the machine halted). |

bits 24–22: opcode

bits 21–19: reg A

bits 18–16: reg B *(unused for S-type and O-type)*

bits 15–0: offsetField or destReg *(unused for O-type)*

**All unused bits in the instruction are set to zero.**

a) Translate the following LC3K assembly code to binary and hexadecimal machine code:

|  |  |  |
| --- | --- | --- |
| **LC3K** | **Binary machine code** | **Hex machine code** |
| bgt 6 7 8 |  |  |
| shift 1 -2 |  |  |

b) Assume that there are n distinct random numbers in memory starting at memory location arr. Write an LC3K **Bubble Sort** so that when your program halts, the numbers in arr are in strictly increasing order. *All registers are initialized to 0.*

|  |  |  |
| --- | --- | --- |
| **C-code** |  | **LC3K assembly** |
| //n is length of arr-1  //n --> R2  i = 0;//i --> R3  do{  j = 0;//j --> R4  status = 0;//status --> R5  do {  //arr[j] --> R6  //arr[j+1] --> R7  if(arr[j] >= arr[j+1])  {  swap(&arr[j],  &arr[j+1]);  status++;  }  j++;  } while (j < n-i);  if (status == 0)  break;  i++;  } while (i < n-1); | 0  1  2  3  4  5  6  7  8  9  10  11  12  13  14  15  16  17  18  19  20  21  22  23  24  25  26  27  28  29  30  31 | lw 0 1 one  lw 0 2 nsub1  outer lw 0 4 zero  lw 0 5 zero  inner \_\_\_\_ 4 6 arr  add 1 4 7  \_\_\_\_ 7 7 arr  bgt \_\_ \_\_ skip  \_\_\_\_ 6 \_\_  sw 4 6 arr  add 1 4 6  sw 6 7 arr  add 1 5 5  skip add 1 4 4  //next 3 insts compute n-i  \_\_\_\_ 3 3 6  add 1 6 6  add 2 6 6  bgt 6 4 \_\_\_\_\_\_  bgt 5 0 \_\_\_\_\_\_  bgt 1 0 end  inc add 1 3 3  bgt 2 3 outer  end halt  zero .fill 0  one .fill 1  nsub1 .fill 4  arr .fill 2  .fill 4  .fill 6  .fill 0  .fill 1 |

## W18: LIMB ISA Points: \_\_\_/ 25

We have come up with a new ISA called LIMB. LIMB is a stack-based, byte-addressable ISA with 16-bit instructions. Registers hold 16 bits of data. The stack is separate hardware structure, each entry is 32 bits. Three instruction types are detailed below. All offsets are two’s complement numbers.

|  |  |  |  |
| --- | --- | --- | --- |
| **S-type** | **15-12: Opcode** | **11-8: regA** | **7-0: signed offset** |

|  |  |  |
| --- | --- | --- |
| Instruction | Opcode | Description |
| breq | 0011 | If the value at the top of the stack == the contents of regA, branch to PC + 2 + offset |
| ldpsh | 0100 | Load word to the top of the stack from memory[regA + offset] (push value of loaded word) |
| store | 0101 | Store the value at the top of the stack into memory at address regA + offset |
| addi | 0110 | Add offset (immediate value) to the contents of regA, putting the result back into regA |

|  |  |  |  |
| --- | --- | --- | --- |
| **Z-type** | **15-12: Opcode** | **11-8: regA** | **7-0: unused** |

|  |  |  |
| --- | --- | --- |
| Instruction | Opcode | Description |
| add | 0111 | Add the value at the top of the stack to the contents of regA, **store in regA** |
| pop | 1000 | Pop value at the top of the stack and put it in regA |
| push | 1001 | Push the contents of regA to the top of the stack |
| cmpz | 1010 | Compare contents of regA with 0. Set appropriate flags for branch (b.lt = branch if less than 0, b.gt = branch if greater than 0, b.eq = branch if equal to 0) |

|  |  |  |
| --- | --- | --- |
| **A-type** | **15-12: Opcode** | **11-0: unused** |

|  |  |  |
| --- | --- | --- |
| Instruction | Opcode | Description |
| halt | 1111 | Stop the program |

**a)** Fill in the following table to convert the instructions to machine code in **Hexadecimal**. Unused bits are set to 0.

|  |  |
| --- | --- |
| Instruction | Machine Code (Hexadecimal 0x) |
| store 7 7 |  |
| ldpsh 2 7 |  |
| push 4 |  |

**b)** For the following program, **assume register 1 contains value 100** and that memory is organized in a **little endian** manner. What are the resulting stack and memory layouts?

0. ldpsh 1 0 // load word to stack from address 100

1. ldpsh 1 4 // load word to stack from address 104

2. ldpsh 1 8 // load word to stack from address 108

3. store 1 0 // store word to memory address 100

Each **stack entry is 32 bits**. Write your **answers in hexadecimal**.

|  |  |  |  |  |
| --- | --- | --- | --- | --- |
| Stack |  | Memory Address | Value | New Value |
|  |  | 100 | 0xAB |  |
|  |  | 101 | 0x10 |  |
|  | (Initial top of stack) | 102 | 0x33 |  |
|  |  | 103 | 0xFB |  |
|  |  | 104 | 0x12 |  |
|  |  | 105 | 0x49 |  |
|  |  | 106 | 0x6D |  |
|  |  | 107 | 0xEE |  |
|  |  | 108 | 0x8C |  |
|  |  | 109 | 0x78 |  |
|  |  | 110 | 0x77 |  |
|  |  | 111 | 0x4D |  |

**c)** The branch instruction and conditions are defined below:

|  |  |
| --- | --- |
| **15-12: Opcode** | **11-0: Offset** |

|  |  |
| --- | --- |
| Instruction | Description |
| b | Unconditionally branch to PC + 2 + Offset |
| b.lt | Branch to PC + 2 + Offset if “cmpz regA” instruction results in “less than”  (reg[regA] < 0) |
| b.gt | Branch to PC + 2 + Offset if “cmpz regA” instruction results in “greater than” (reg[regA] > 0) |
| b.eq | Branch to PC + 2 + Offset if “cmpz regA” instruction results in “equal to”  (reg[regA] == 0) |

Say we have an array Arr[ ] of 4 integers (int32\_t). At the end of the following LIMB program, the stack will contain the values from Arr[ ] that are negative. Complete the following program. **Assume all registers contain 0 initially and Arr[] starts at address 100.**

//pseudo code

for (int i = 3; i >= 0; i--) {

if (Arr[i] < 0) {

pushToStack(Arr[i]);

}

}

0. addi 2 **\_\_** //initialize loop counter

1. loop **\_\_\_\_\_** 2

2. b.lt **\_\_\_\_\_\_**

3. **\_\_\_\_\_** 2 100 // load word to stack from Arr[i]

4. pop **\_\_** // put value in reg 3

5. cmpz 3

6. b.lt **\_\_\_\_\_\_**

7. dec addi 2 **\_\_** // else, decrease address and branch

8. b loop

9. pstack push 3 //push negative element to stack

10. b **\_\_\_\_\_\_**

11. end halt

## W21: New ISA

To improve on LC2K, we’ve developed a new, modern ISA called the LCJINX with 12 opcodes and 16 registers. Unfortunately, since our design team got lazy, we can still only support 16 bit signed immediate values. (note: Jinx definitely did not write this question)

There are four types of instructions:

* J-type instructions take in three operands, which are all registers.
* I-type instructions take in three operands, where two are registers and the last is an immediate value.
* N-type instructions take in no operands.
* X-type instructions take in four operands, where all four operands are registers.

1. **What is the minimum word size of the LCJINX? Round your answer up to the nearest byte. (3 points)**

Assume the LCJINX is word-addressable, with a word size of 8 bytes. The instructions for the LCJINX are as follows:

|  |  |  |  |
| --- | --- | --- | --- |
| **Instruction** | **Type** | **Opcode** | **Description** |
| add rA rB rD | J | 0x0 | Adds the values in registers A and B and stores the result to register D (destination). |
| nand rA rB rD | J | 0x1 | Takes the bitwise NAND of the values in registers A and B and stores the result to register D (destination). |
| addi rA, rD, imm | I | 0x2 | Adds the value in register A with the immediate value and stores the result to register D (destination). |
| sft rA, rB, imm | I | 0x3 | Takes the value in register A and shifts it by the number of bits specified by *imm*. If *imm* is positive, then the value is shifted to the right by *imm* bits. If *imm* is negative, the value is shifted to the left by |*imm*| bits. The result is stored into register B. |
| ld rA, rB, imm | I | 0x4 | Loads the value in memory at address (value of register A + immediate value) to register B. |
| str rA, rB, imm | I | 0x5 | Stores the value in register B to memory address (value of register A + immediate value). |
| beq rA, rB, imm | I | 0x6 | If the values in rA and rB are the same, then branch to PC = *imm*. (note: different from LC2K’s beq!) |
| blt rA, rB, imm | I | 0x7 | No, not the sandwich. If the value in register A is less than (<) the value in register B, then branch to PC = *imm*. |
| meow | N | 0x8 | Equivalent to LC2K’s *noop* instruction. Does nothing. |
| halt | N | 0x9 | Equivalent to LC2K’s *halt* instruction. Halts the program. |
| cmov rA, rB, rC, rD | X | 0xA | If the values of register A and register B are equal, set the value of register C to be the value in register D. |
| beqlr rA, rB, rC, rD | X | 0xB | If the values of register A and register B are equal, store the current PC+1 into register D and branch to the value in register C. This is similar to LC2K’s *jalr* instruction. |

For backwards compatibility and ease of use, the *.fill* directive works exactly the same as LC2K. Labels have a maximum length of 12 characters, for the optimal programmer experience (or as much as you can get while writing in assembly). All symbolic addresses resolve to the memory address of that label. Assume all registers start with the value 0.

**J-type instructions are encoded as follows:**

**bits 35-32: opcode**

**bits 31-28: register A**

**bits 27-24: register B**

**bits 23-4: unused (all 0)**

**bits 3-0: register D**

**I-type instructions are encoded as follows:**

**bits 35-32: opcode**

**bits 31-28: register A**

**bits 27-24: register B**

**bits 23-16: unused (all 0)**

**bits 15-0: 16-bit immediate**

**N-type instructions are encoded as follows:**

**bits 35-32: opcode**

**bits 31-0: unused (all 0)**

**X-type instructions are encoded as follows:**

**bits 35-32: opcode**

**bits 31-28: register A**

**bits 27-24: register B**

**bits 23-20: register C**

**bits 19-16: register D**

**bits 15-0: unused (all 0)**

1. **Complete the machine code translation of the LCJINX code shown below. Express your answers in hex.**

**Assuming all registers start as 0 at the beginning of the program, what is the value in register 5? (7 points, 4.5 for MC and 2.5 for register)**

**LCJ1NX Assembly Machine Code**

addi 1 1 16 0x211000010

sft 1 2 3 0x312000003

nand 0 0 3 0x100000003

add 2 3 4 0x\_\_\_\_\_\_\_\_\_

sft 4 4 -4 0x\_\_\_\_\_\_\_\_\_

cmov 1 4 5 3 0x\_\_\_\_\_\_\_\_\_

halt 0x900000000

1. **Your team has decided to use the 370-developed J1 chip to run LCJINX, and you’re now tasked with a program to run binary search in a sorted array. Your teammates had a working prototype, but a sleep- and caffeine-deprived programmer accidentally deleted a few lines of code. Help restore the program below based on the comments. (10 points)**

ld 0 14 searchAddr

beqlr 0 0 14 15 // call binary search function

halt

meow

searchFunc add 0 0 1 // l = 0

ld 0 2 arraySize // r = array size

ld 0 9 target // load target

loop beq 1 2 nofind // if l == r, not found

\_\_\_\_ \_ \_ \_\_\_\_\_\_ // find m = (l+r) / 2 (2 lines)

\_\_\_\_ \_ \_ \_\_\_\_\_\_ // store m into **reg 3**

\_\_\_\_ \_ \_ \_\_\_\_\_\_ // load array[m] into **reg 5**

beq \_ \_ \_\_\_\_\_\_ // if array[m] == target

\_\_\_\_ \_ \_ \_\_\_\_\_\_ // handle inequalities

\_\_\_\_ \_ \_ \_\_\_\_\_\_ // else

\_\_\_\_ \_ \_ \_\_\_\_\_\_ // else

equal add 0 3 13 // return m in **reg 13**

beq 0 0 done // done!

\_\_\_\_\_\_ \_\_\_\_ \_ \_ \_\_\_\_\_\_ // inequality case 1

\_\_\_\_ \_ \_ \_\_\_\_\_\_ // inequality case 1

done beqlr 0 0 15 14 // return

nofind \_\_\_\_ \_ \_ \_\_\_\_\_\_ // load return value with -1

beqlr 0 0 15 14 // return

searchAddr .fill searchFunc

arraySize .fill 8

target .fill 7

arrayAddr .fill array

array .fill 1

.fill 3

.fill 4

.fill 7

.fill 10

.fill 12

.fill 13

.fill 16

## W22: New ISA [12 Points]

To cut hardware costs, a team of engineers has developed a simplified version of the LC2K ISA with only four registers—**buffer**, **RZR**, **RPR**, and **RSC**—which have the following functionality:

|  |  |
| --- | --- |
| **Register** | **Description** |
| buffer | Used to move register values to/from memory via the instructions **loadbuf** and **storebuf**, or between different registers via the instructions **loadreg** and **storereg** |
| R**ZR** (“**Z**e**r**o Reg”) | Holds the value 0 and cannot be changed |
| R**PR** (“**Pr**imary Reg”) | Holds values for compute, akin to standard LC2K registers |
| R**SC** (“**S**e**c**ondary Reg”) | Holds values for compute, akin to standard LC2K registers |

This new ISA[[1]](#footnote-0) has the following instructions and semantics:

|  |  |  |
| --- | --- | --- |
| **Assembly Code** | **Execution  Semantics** | **Usage Example** |
| addi **imm.** | **RPR** = **RPR** + **imm.** (Add **imm.** value to **RPR**) | addi 10 |
| add | **RPR** = **RPR** + **RSC** | add (**No args**) |
| nandi **imm.** | **RPR** = **RPR** NAND **imm.** (Nand **imm.** value with **RPR**) | nandi 42 |
| nand | **RPR** = **RPR** NAND **RSC** | nand (**No args**) |
| beq **label** | Branch to **label** if **RPR** == **RSC** | beq end\_label |
| loadreg **reg** | **reg** = **buffer** (**reg** can be **RPR** or **RSC**) | loadreg RPR |
| storereg **reg** | **buffer** = **reg**  (**reg** can be **RPR**, **RSC**, or **RZR**) | storereg RZR |
| loadbuf **imm.** | **buffer** = mem[**imm.**] (load from **imm.** address) | loadbuf 32 |
| storebuf **imm.** | mem[**imm.**] = buffer (store to **imm.** address) | storebuf 64 |

|  |  |
| --- | --- |
| **(a)** | **[2 pts]** If each instruction is encoded in 16 bits, and the addresses in **loadbuf** and **storebuf** instructions are unsigned immediates, what is the maximum number of address bits that can be in this ISA? (Assume opcodes are the same size in all instructions) What is the highest address a load or store instruction could refer to? |

Maximum number of address bits: \_\_\_\_\_\_\_\_\_\_\_\_\_\_

Highest loadable/storable address: \_\_\_\_\_\_\_\_\_\_\_\_\_\_

|  |  |
| --- | --- |
| **(b)** | **[5 pts]** Use the new ISA to implement the C pseudocode on the left. Solutions exist that don’t use all lines. The variable mem is an array of 32-bit values, thus each index corresponds to a single memory address. (*Hint*: NAND gates can create AND gates as shown in this diagram): |

![](data:image/png;base64,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)

|  |  |
| --- | --- |
| **C Pseudocode** | **New ISA** |
| if (mem[8] & 1) {  Branch to label “goal”  } | loadbuf 8  loadreg RPR  \_\_\_\_\_\_\_\_\_\_\_  \_\_\_\_\_\_\_\_\_\_\_  \_\_\_\_\_\_\_\_\_\_\_  \_\_\_\_\_\_\_\_\_\_\_  storereg RZR  \_\_\_\_\_\_\_\_\_\_\_  beq end  \_\_\_\_\_\_\_\_\_\_\_  beq goal  end |

|  |  |
| --- | --- |
| **(c)** | **[5 pts]** Use the new ISA to implement the C pseudocode on the left. Solutions exist that don’t use all lines. The variable mem is an array of 32-bit values, thus each index corresponds to a single memory address. |

|  |  |
| --- | --- |
| **C Pseudocode** | **New ISA** |
| mem[100] = (mem[64] + 3) << 2; | loadbuf 64  loadreg RPR  \_\_\_\_\_\_\_\_\_\_\_  \_\_\_\_\_\_\_\_\_\_\_  \_\_\_\_\_\_\_\_\_\_\_  \_\_\_\_\_\_\_\_\_\_\_  \_\_\_\_\_\_\_\_\_\_\_  \_\_\_\_\_\_\_\_\_\_\_  \_\_\_\_\_\_\_\_\_\_\_  storereg RPR  storebuf 100 |

1. A copy of ISA can be found in the supplemental material for ease of reference. [↑](#footnote-ref-0)